Model fitting exercise

Step 1

Create a diagram named X and run the following commands to create a simple model configuration:

aadd "X" "POINT" "PO1" (100,110)
aadd "X" "POINT" "PO2" (140,110)
aadd "X" "POINT" "PO3" (180,100)
aadd "X" "POINT" "PO4" (180,120)
aadd "X" "PIPE" "PI1" (120,110)
aadd "X" "PIPE" "PI2" (160,100)
aadd "X" "PIPE" "PI3" (160,120)
amodi "PI1" "PI12_CONNECT_POINT_1" "PO1"
amodi "PI1" "PI12_CONNECT_POINT_2" "PO2"
amodi "PI2" "PI12_CONNECT_POINT_1" "PO2"
amodi "PI2" "PI12_CONNECT_POINT_2" "PO3"
amodi "PI3" "PI12_CONNECT_POINT_1" "PO2"
amodi "PI3" "PI12_CONNECT_POINT_2" "PO4"
aexclude "PO1"
aexclude "PO3"
aexclude "PO4"
amodi "PI1" "PI12_AREA" 0.06
amodi "PI2" "PI12_AREA" 0.03
amodi "PI3" "PI12_AREA" 0.03
amodi "ECCO" "MAXIMUM_TIME_STEP" 0.05
amodi "ECCO" "CURRENT_TIME_STEP" 0.05
amodi "SPEED" "SC_SPEED" 1000.0

Save the initial condition and set the simulation time to zero.

Step 2

Run the following commands to print how mass flow of the pipe PI1 behaves in a simple transient.

loadInitialCondition (syncRead $ \_ -> fromResource $ relativeResource currentModel "/Initial%20Condition")
runSequence mdo
      execute (setVar "PO1#PO11_PRESSURE" 1.1)
      fork (wait 1 >> stop)
      fork $ repeatForever mdo
          wait 0.05
          execute do
              massFlow = getVar "PI1#PI12_MIX_MASS_FLOW" :: Double
              print massFlow

Step 3

Modify the above commands so that they print the index of the time step and variables PI1#PI12_MIX_MASS_FLOW, PI2#PI12_MIX_MASS_FLOW and PI3#PI12_MIX_MASS_FLOW. The commands should produce something like:

0 3.4978646673069216 1.728445205173922 1.728445205173922
1 143.55517382976987 71.78198788751189 71.78198788751189
2 261.95050765748715 130.97625819564252 130.97625819564252
3 354.3858379588669 177.19356818048865 177.19356818048865
...

Use the following functions to maintain the counter (although it is possible to compute it also from the current simulation time).

ref :: a -> <Proc> Ref a (Prelude)

Creates a new reference with the given initial value.

getRef :: Ref a -> <Proc> a (Prelude)

Returns the current value of the reference.

(:=) :: Ref a -> a -> <Proc> () (Prelude)

Sets a new value for the reference.

Step 4

Modify the commands so that they compute the squared sum of errors from the "measurements" below:

measurements = [
    [0.05, 3.4977860762417605, 1.7283740169550754, 1.7284367287300313],
    [0.1, 134.88381983045093, 64.08353845423913, 70.80714606160574],
    [0.15, 231.9062548984293, 105.65792385470164, 126.24691709056889],
    [0.2, 296.4661748497294, 130.5064306373064, 165.95821168905044],
    [0.25, 336.98201115920403, 144.68635329210002, 192.29464808262938],
    [0.3, 361.56600347367186, 152.64673170010516, 208.9187479324867],
    [0.35, 376.0973084183755, 157.07188101607773, 219.0252113124207],
    [0.4, 384.690502383305, 159.57004084578205, 225.12040809039868],
    [0.45, 389.742925383871, 160.98719663407357, 228.75573737254442],
    [0.5, 392.7040735842458, 161.79511170873698, 230.90898615966157],
    [0.55, 394.4365833534546, 162.2577698637119, 232.17883313521224],
    [0.6, 395.36535200557034, 162.50160633728285, 232.86375430699712],
    [0.65, 395.93996807514526, 162.65081127152007, 233.28915819466746],
    [0.7, 396.2954901434386, 162.74229076331505, 233.55318938336018],
    [0.75, 396.51546801100426, 162.79847242004152, 233.71697886953123],
    [0.8, 396.65158867682203, 162.83302418223676, 233.81854447407886],
    [0.85, 396.7358254901475, 162.85429827014457, 233.881506261103],
    [0.9, 396.7879581657287, 162.8674097925786, 233.92052799957008],
    [0.95, 396.8202241096274, 162.87549715428165, 233.94470806962704],
    [1.0, 396.8401951361238, 162.88048891915273, 233.95968928259308]]

The columns of the data are, the simulation time PI1#PI12_MIX_MASS_FLOW, PI2#PI12_MIX_MASS_FLOW and PI3#PI12_MIX_MASS_FLOW.

Step 5

Now create a function objFun :: [Double] -> <Proc> Double that is called as

objFun [1.0,1.1,1.2]

It should run the same commands as in previous steps and return the squared sum of errors, but additionally set PI1#PI12_LOSS_COEFF, PI2#PI12_LOSS_COEFF and PI3#PI12_LOSS_COEFF to the given values.

It is good idea to also print the parameter values at the beginning of the function (and remove all other printing).

Step 6

Use the function

to fit the loss coefficients to the measurements. Try first with small number of function evaluations, because the current implementation does not allow interrupting the optimization. You may try rhobeg=1 and rhoend=1e-4 and initial guess [1,1,1].